Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Mark reactions as IMAP-seen in marknoticed_chat() (#6210) #6213

Open
wants to merge 2 commits into
base: main
Choose a base branch
from

Conversation

iequidoo
Copy link
Collaborator

@iequidoo iequidoo commented Nov 15, 2024

See commit messages.
Close #6210

For DC_CHAT_ID_ARCHIVED_LINK this does nothing, marking all reactions as seen in the archive doesn't seem useful. mark_old_messages_as_noticed() only marks reactions as InNoticed, this function is called from imap when new messages arrive, so it is anyway called on all devices, no synchronisation is needed.

DONE:

@iequidoo iequidoo marked this pull request as ready for review November 15, 2024 23:41
@iequidoo iequidoo requested review from r10s and Hocuri November 16, 2024 01:08
src/chat.rs Show resolved Hide resolved
@iequidoo iequidoo marked this pull request as draft November 16, 2024 17:09
@Hocuri
Copy link
Collaborator

Hocuri commented Nov 18, 2024

Since you marked it as a draft, I take it that it's not ready to be reviewed yet?

@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from a089199 to b71c681 Compare November 18, 2024 18:58
@iequidoo iequidoo marked this pull request as ready for review November 18, 2024 19:00
@iequidoo
Copy link
Collaborator Author

iequidoo commented Nov 18, 2024

Since you marked it as a draft, I take it that it's not ready to be reviewed yet?

Now it's ready and seems working, but probably needs more tests. Was a draft because i needed to optimise out saving unnecessary columns to db.

It's interesting that sent out reactions were already hidden messages, didn't know about that.

@Hocuri Hocuri requested a review from link2xt November 18, 2024 19:22
@Hocuri
Copy link
Collaborator

Hocuri commented Nov 19, 2024

Ideally there will be a test that marks a reaction as seen on one device and checks that a MsgsNoticed event is emitted on another device (and check that the test fails without this PR here), in addition to the unit tests. If this is not feasible, would be good to manually test that this event is sent.

Edit: If, during testing, it turns out that it's harder to do than expected, it would be fine to just ignore the issue, see deltachat/deltachat-android#3377 (comment)

@iequidoo
Copy link
Collaborator Author

iequidoo commented Nov 21, 2024

If, during testing, it turns out that it's harder to do than expected, it would be fine to just ignore the issue, see deltachat/deltachat-android#3377 (comment)

I think it's not so difficult to write a jsonrpc Python test for that and it would be good to have incoming reactions in the msgs table samely as it's for outgoing reactions (just for uniformity, though this can be done w/o marking reactions as IMAP-seen), so i will try to add a test. EDIT: Done.

@iequidoo iequidoo marked this pull request as draft November 22, 2024 01:33
@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch 3 times, most recently from de2b5a1 to d9482dc Compare November 22, 2024 01:55
@iequidoo iequidoo marked this pull request as ready for review November 22, 2024 02:12
Copy link
Collaborator

@Hocuri Hocuri left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Nice that not a lot of added complexity was necessary

src/chat.rs Outdated Show resolved Hide resolved
deltachat-rpc-client/tests/test_something.py Show resolved Hide resolved
Comment on lines 1020 to 1021

state = if seen
|| fetching_existing_messages
|| is_mdn
|| is_reaction
|| chat_id_blocked == Blocked::Yes
{
state = if seen || fetching_existing_messages || is_mdn || chat_id_blocked == Blocked::Yes {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This change leads to IncomingMsg instead of MsgsChanged being emitted, so that the UI shows a notification "Hocuri: ❤️" if I reacted with a heart emoji. Which we don't want, since we instead to show the "Hocuri reacted ❤️ to "I like your avatar!"" message in the UI.

So, this needs to be:

         state = if seen || fetching_existing_messages || is_mdn || chat_id_blocked == Blocked::Yes {
             MessageState::InSeen
         } else if is_reaction {
             MessageState::InNoticed
         } else {
             MessageState::InFresh
         };

Copy link
Collaborator Author

@iequidoo iequidoo Nov 23, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, this is a bug which requires a separate test it seems.

But again, making reactions InNoticed to only make marknoticed_chat() faster doesn't look conceptually correct. Reactions are still a user-noticeable info, so there's smth to "notice". And wrt the bug you found, rather we shouldn't emit IncomingMsg for hidden messages. But not sure, need to try and see if the tests doesn't break at least. EDIT: Tried, the Rust tests continue to work at least.

Copy link
Collaborator Author

@iequidoo iequidoo Nov 24, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

A PR extending the existing tests to catch this bug: #6257. Checked that the modified tests fail for this PR

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Incoming reactions aren't really InFresh either, because InFresh means that the message is counted in the unread-counter and is shown in notifications. And info messages are also immediately added as InNoticed, because that's what they semantically are.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not necessarily, get_fresh_msg_cnt() includes the and m.hidden=0 SQL condition, so if we need, we can make incoming reactions hidden InFresh messages. As for "is shown in notifications", now this is true for incoming reactions, that's also why i think they should be InFresh. If we add them as InNoticed and show notifications for them, that would be different from incoming messages. So, my suggestion is:

  • Add incoming reactions as as hidden InFresh messages.
  • Add other incoming hidden messages as InSeen. But looking at receive_imf.rs i don't see such messages even exist. But then i don't understand why we have and m.hidden=0 checks for incoming messages in SQL everywhere. So this needs investigation.
  • Add a db migration for existing hidden InFresh messages. At least just in case.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

From a code perspective, we can make them InFresh, it requires one more special case (in order not to send an IncomingMsg), but that doesn't make any difference, really.

What does make a difference is whether we introduce 1ms of delay (1ms on quite a new phone, that is - older phones may easily need 3 times as long). Doing this just once isn't noticeable, but if we do it often (and without necessarily needing to) then we will end up with slow code (and markseen is called every time a chat is opened). Measuring this isn't hard, you can also do it - if you find a way that is just as fast, I'm fine with that, too. You may need to measure on a phone, not on Desktop, because on a fast PC the difference may not be noticeable.

Add a db migration for existing hidden InFresh messages. At least just in case.

We shouldn't do database migrations "just in case", since there is always some chance that they create problems, and we don't have any indication that it will solve problems.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

From a code perspective, we can make them InFresh, it requires one more special case (in order not to send an IncomingMsg)

Done in 110f0d3, doesn't look complex.

What does make a difference is whether we introduce 1ms of delay

I'm going to profile that, but anyway, there's a new SQL statement selecting hidden messages (actually reactions), so i'm afraid it will always add some delay (NB: message::markseen_msgs() is a no-op if msg_ids is empty, so it's not a source of the problem). In #6213 (comment) you suggested a solution:

This can be fixed by writing WHERE state=? here and replacing MessageState::InFresh, MessageState::InSeen with MessageState::InNoticed below.

but i don't understand how it works -- just because you simplify the SQL statement (which is unlikely) or because you don't select other unnecessary messages and don't pass them to message::markseen_msgs()? But there should be no other hidden InFresh messages currently (NB: Only looking at current receive_imf.rs, i'm not sure they were never created in the past).

We shouldn't do database migrations "just in case", since there is always some chance that they create problems

See "NB" above -- if hidden InFresh messages were created in the past, they will be selected and passed to message::markseen_msgs() which we should avoid. So, probably this db migration will be a no-op, but as we can't be sure (we can't inspect all the previously released versions, really), better to add it.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

but i don't understand how it works -- just because you simplify the SQL statement (which is unlikely) or because you don't select other unnecessary messages and don't pass them to message::markseen_msgs()?

I don't know, my guess would be that it's because the simplification in the SQL statement enabled some optimization.

Copy link
Collaborator Author

@iequidoo iequidoo Nov 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe it works because we have msgs_index7 ON msgs (state, hidden, chat_id). If so, i should replace state>=? AND state<? with an OR condition containing just InFresh and InNoticed.

EDIT: Maybe just with state=<InFresh> because InNoticed reactions are normally a result of mark_old_messages_as_noticed() which is called from imap when new messages arrive, but that doesn't need synchronisation using the \Seen flag because that happens on all devices anyway.

EDIT: This will lead to not all reactions marked as seen on IMAP, but i think it's ok, better to optimise here.

@Hocuri
Copy link
Collaborator

Hocuri commented Nov 25, 2024

BTW, this should be changed now:

// Let hidden messages also be ordered with protection messages because hidden messages
// also can be or not be verified, so let's preserve this information -- even it's not
// used currently, it can be useful in the future versions.

We have hidden messages now, but they shouldn't affect how other messages are ordered, so AND hidden=0 should be added.

@iequidoo
Copy link
Collaborator Author

BTW, this should be changed now:

// Let hidden messages also be ordered with protection messages because hidden messages
// also can be or not be verified, so let's preserve this information -- even it's not
// used currently, it can be useful in the future versions.

We have hidden messages now, but they shouldn't affect how other messages are ordered, so AND hidden=0 should be added.

This is a code path for protection messages. Protection messages should be sorted under any hidden messages, incl. reactions, and we already have outgoing reactions as hidden messages. The comment just says why we don't add AND hidden=0 here, maybe it should be improved?

As for the else if received { code block, AND hidden=0 is already there because of the reason you pointed to.

@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch 2 times, most recently from 110f0d3 to 3e8b050 Compare November 26, 2024 20:13
src/chat.rs Outdated
let mut stmt = conn.prepare(
"SELECT id FROM msgs
WHERE state=? AND hidden=1 AND chat_id=?
ORDER BY id DESC LIMIT 100",
Copy link
Collaborator Author

@iequidoo iequidoo Nov 26, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now this statement should be fast as we have msgs_index7 ON msgs (state, hidden, chat_id), but maybe we want LIMIT 1 here because the effect is the same anyway. This is what i don't understand:

  • Suppose one device received reaction A.
  • And another device -- A and B.
  • The user clicks on "Mark as read" on the first device.
  • On the second device MsgsNoticed is emitted which results in removing a notification for reaction B.
  • Then the user has a chance to see the notification for reaction B on the first device only.

CC @r10s

Seems the same problem exists for "usual" messages because MsgsNoticed only contains ChatId. Didn't know about that. I hope this is just my misunderstanding.

EDIT: But if i'm right and there were no user complaints for years, this isn't a big problem and may be solved later. I changed the LIMIT in the SQL to 1, this doesn't really matter. Some reactions will be marked as seen, some won't, this isn't important.

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Did you measure it?

Copy link
Collaborator Author

@iequidoo iequidoo Nov 28, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This SELECT is already fast enough, but i added another commit to optimize the UPDATE statement. On my machine for some random chat now it takes 59.086µs vs 41.202µs w/o this PR, so i think the degradation is acceptable. The whole function took 231.989µs. So, even though the hidden value set is [0, 1], it must be checked explicitly in statements to employ the corresponding db index.

@iequidoo iequidoo marked this pull request as draft November 27, 2024 23:19
@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from 3e8b050 to 4e257c5 Compare November 28, 2024 04:23
@iequidoo iequidoo marked this pull request as ready for review November 28, 2024 04:24
@iequidoo
Copy link
Collaborator Author

iequidoo commented Nov 28, 2024

Removed the unnecessary db migration because even if InFresh hidden messages exist currently, now they might affect the logic only once resulting in marking the last such message as \Seen on IMAP. The performance issue of the SQL in marknoticed_chat() is solved by not using inequalities on state to employ msgs_index7 ON msgs (state, hidden, chat_id) there. Inequalities should only be used for the last column of the index, i.e. chat_id which may make sense to filter out special chats. So now i think this is ready for review.

@iequidoo iequidoo requested a review from Hocuri November 28, 2024 04:38
@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from 4e257c5 to 27a1244 Compare November 28, 2024 20:59
@iequidoo iequidoo requested a review from Hocuri December 6, 2024 17:03
@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from 27a1244 to 92e8898 Compare December 6, 2024 17:26
@iequidoo
Copy link
Collaborator Author

iequidoo commented Dec 7, 2024

@Hocuri An alternative to LIMIT 1 (only marking the last reaction as seen) may be e.g. LIMIT 100 and marking all reactions as seen eventually (just for uniformity), but then we need to tweak the other (synchronized) side to only emit MsgsNoticed when fresh reactions have been seen on the first device. Some old reactions seen on another device isn't smth interesting.

@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch 2 times, most recently from 6c90599 to 3aa1e03 Compare December 12, 2024 19:20
@iequidoo iequidoo marked this pull request as draft December 13, 2024 15:28
@link2xt
Copy link
Collaborator

link2xt commented Dec 13, 2024

Currently (without this PR) reactions are never marked as seen on IMAP, right?

src/receive_imf.rs Outdated Show resolved Hide resolved
@Hocuri
Copy link
Collaborator

Hocuri commented Dec 13, 2024

Currently (without this PR) reactions are never marked as seen on IMAP, right?

Yes

An alternative to LIMIT 1 (only marking the last reaction as seen) may be e.g. LIMIT 100 and marking all reactions as seen eventually (just for uniformity), but then we need to tweak the other (synchronized) side to only emit MsgsNoticed when fresh reactions have been seen on the first device. Some old reactions seen on another device isn't smth interesting.

The same is true for LIMIT 1 as for LIMIT 100? But anyway, MsgsNoticed is only used to remove notifications, so that a MsgsNoticed for a chat without fresh messages doesn't have any effect (because there are no notifications to remove).

@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from 3aa1e03 to 00122b8 Compare December 13, 2024 16:51
@iequidoo iequidoo marked this pull request as ready for review December 13, 2024 16:53
@iequidoo iequidoo marked this pull request as draft December 13, 2024 16:53
@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from 00122b8 to e71b17c Compare December 13, 2024 16:56
@iequidoo iequidoo marked this pull request as ready for review December 13, 2024 16:56
@iequidoo
Copy link
Collaborator Author

I split this into two commit for easier reviewing at least.

@iequidoo
Copy link
Collaborator Author

The same is true for LIMIT 1 as for LIMIT 100? But anyway, MsgsNoticed is only used to remove notifications, so that a MsgsNoticed for a chat without fresh messages doesn't have any effect (because there are no notifications to remove).

The problem is that MsgsNoticed will be emitted on other devices if we mark some old reactions as seen. Another device may have more received messages and useful notifications, and they will be removed then (this is possible anyway, but we would increase probability of this). That's why i decided not to mark old reactions as seen, moreover, it's an extra work on IMAP.

@iequidoo iequidoo requested a review from link2xt December 13, 2024 17:22
@Hocuri
Copy link
Collaborator

Hocuri commented Dec 19, 2024

After digging through the PR for a bit more, I now understand the new logic to be:

  • marknoticed_chat() and mark_old_messages_as_noticed() are the only two functions marking reactions as InNoticed.
  • After mark_old_messages_as_noticed() was called, it's not necessary anymore to mark reactions as Seen on IMAP because it's triggered by an outgoing message, which will simultaneously trigger mark_old_messages_as_noticed() on other devices, so, reactions will already be removed there.
  • So, marknoticed_chat() SELECTs the last incoming hidden message in the current chat, and only if it's InFresh will it call markseen_msgs(). This is correct because the only reason for it not to be InFresh is that either marknoticed_chat() or mark_old _messages_as_noticed() was called since the message was received, and either of these remove the reactions from other devices.

However, it took me quite long to understand this logic even though I know the surrounding code quite good right now, so, for someone who doesn't know the surrounding code it will be super hard.

Moreover, the increased complexity doesn't make everything perfect, e.g. consider the following scenario:

  • Device A goes offline
  • Device B receives 2 different reactions (either from different people or on different messages) in the same chat.
  • The chat is opened on device B.
  • -> Device B marks the second reaction as seen; the first reaction is untouched because of LIMIT 1
  • Device A goes online.
  • -> Device A shows a notification for the first reaction, even though the user already saw it on device B.

So, in order to a) reduce the complexity and b) fix this problem and c) slightly improve the performance of marknoticed_chat() (because SQL will need to do a bit less work), I still suggest the following logic:

  • Reactions will only ever be InNoticed or InSeen; they will never be InFresh. (the logic would also work if we make them InFresh - in this case, there is slightly more complexity, but not a lot and at least it's not complicated).
  • marknoticed_chat() marks all hidden InNoticed messages in the current chat as Seen by calling markseen_msgs(). Put a comment that this is for reaction messages, and reaction messages are never InFresh, so it's enough to check for InNoticed.

@iequidoo
Copy link
Collaborator Author

iequidoo commented Dec 20, 2024

-> Device A shows a notification for the first reaction, even though the user already saw it on device B.

Good catch, but this is a bug that already exists in the current code for "usual" messages because sync_seen_flags() emits MsgsNoticed(chat_id) for existing messages seen on other devices, but receive_imf doesn't do so when it receives a seen message. So, the user may see a new message notification, just swipe it, then see another new message notification and mark it as read, and when their second device goes online, it will show a notification for the first message, and it won't be removed because MsgsNoticed isn't emitted. I have even checked this with my DC Android and Desktop. receive_imf should emit MsgsNoticed for InSeen messages, this is obviously forgotten. If this is fixed (EDIT: Fixed in a separate PR #6351), returning to your scenario, the notification on device A will be quickly auto-removed because right after that device A receives MsgsNoticed triggered by the second reaction seen on device B. To avoid notifications flickering (which is also possible with "usual" messages currently), UIs should probably postpone showing them until all events are read. But idk how easy this is, maybe we should postpone solving this problem.

a) reduce the complexity

The first commit doesn't add any complexity, it rather unifies things for all reactions, so let's talk about feat: Mark reactions as IMAP-seen in marknoticed_chat(). The are several things adding complexity, just for the record:

  • struct ReceivedMsg gains a new hidden field. It's needed because incoming reactions are InFresh and we don't emit IncomingMsg for them. This can be fixed by adding incoming reactions as InNoticed as you suggested.
  • Some SQL statements now need hidden IN (0,1) instead of hidden=0 to change reactions' state as well. Can be fixed the same way. Though i don't think that complicates the code a lot.
  • New SELECT in marknoticed_chat() with Rust filter() expr.

c) slightly reduce the performance of marknoticed_chat() (because SQL will need to do a bit less work)

Probably you want to say "slightly improve". Yes, if incoming reactions are added as InNoticed, SQL becomes a bit simpler (see above) and doesn't need to change hidden messages' state, but only for "visible" ones.

But despite of the all advantages of reactions being added as InNoticed, a conceptual disadvantage is that then marknoticed_chat() transfers visible messages from InFresh to InNoticed, but hidden ones -- from InNoticed to InSeen. I think marknoticed_chat() should only handle InFresh messages and i'm almost sure that handling already InNoticed messages there is bug-prone. Moreover, if we add reactions as InNoticed, mark_old_messages_as_noticed() is no-op for them and it doesn't prevent marknoticed_chat() from marking them (or at least the last reaction) as seen on IMAP which is at least an extra work.

  • marknoticed_chat() marks all hidden InNoticed messages in the current chat as Seen by calling markseen_msgs().

Anyway LIMIT should be added to SQL SELECT which also means we can't mark all such messages as seen in one call to marknoticed_chat(). Then i'd say it's not worth the effort to mark all hidden incoming messages as seen even eventually and deal with multiple MsgsNoticed events on all devices.

Before, received reactions went to the trash chat. Make them hidden chat messages as already done
for sent reactions, just for unification. Incoming received reactions remain `InSeen`, so no changes
are needed to `chat::marknoticed_chat()` et al.
@iequidoo iequidoo force-pushed the iequidoo/seen-reactions branch from c3c64b5 to 53d53ee Compare January 4, 2025 00:02
When a reaction notification is shown in the UIs, there's an option "Mark Read", but the UIs are
unaware of reactions message ids, so the UIs just call `marknoticed_chat()` in this case. We don't
want to introduce reactions message ids to the UIs (at least currently), but let's make received
reactions `InFresh` so that the existing `\Seen` flag synchronisation mechanism works for them, and
mark the last fresh hidden incoming message (reaction) in the chat as seen in
`chat::marknoticed_chat()` to trigger emitting `MsgsNoticed` on other devices.

There's a problem though that another device may have more reactions received and not yet seen
notifications are removed from it when handling `MsgsNoticed`, but the same problem already exists
for "usual" messages, so let's not solve it for now.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Make it possible to mark reactions as "seen"
3 participants